

# 2D Digital Systems Lab Report Pseudo Random Number Generator

prepared by

Stephanie See Hui Ying (1005440)

Lee Jun Lei, Lucas (1006341)

Jarell Lim En Yu (1006350)

Syarifuddin Azhar Bin Rosli (1005187)

Mateo Jalen Andrew Calderon (1006042)

## **Abstract**

This project develops a pseudo-random number generator (PRNG) using the logistic map, implemented in Verilog HDL on an FPGA and processes its output through a Central Limit Theorem (CLT) function to achieve a Gaussian distribution. The system integrates additional FPGA modules for real-time interaction and visualisation, including a clock generator, UART interface, XADC, and a 7-segment display driver. These components facilitate the direct display of PRNG values on the FPGA and the transmission of data to a laptop for histogram analysis, verifying the Gaussian nature of the output. This approach demonstrates the practical application of chaotic systems for generating Gaussian-distributed pseudo-random numbers in digital hardware, highlighting the logistic map's potential in PRNG design.

## **Table of Contents**

| Abstract                          |    |
|-----------------------------------|----|
| Table of Contents                 |    |
| 1. Introduction                   |    |
| 2. Literature Review              |    |
| 3. PRNG Algorithm                 |    |
| 3.1 Proof Of Concept              |    |
| 3.2 FPGA Implementation of PRNG   |    |
| 4. Setup of FPGA                  | 9  |
| Actual Verilog HDL implementation |    |
| 6. FPGA Testbenches               |    |
| 7. Results                        |    |
| 8. Reflections and Challenges     | 17 |
| 9. Group Image                    |    |
| 10. References                    |    |

#### 1. Introduction

Our project set out to create a reliable pseudo-random number generator (PRNG), an essential component in many technological applications today, and implement it using the CMOD-A7. Our interest was piqued by chaotic systems, known for their sensitivity to initial conditions, or "seeds," making them an intriguing choice for PRNGs. Among various chaotic models we explored, including the Lorentz equation and the double pendulum, the logistic map captured our attention due to its straightforward yet chaotic nature, positioning it as the centrepiece of our investigation.

The logistic map's appeal lies in its deterministic behaviour that, when initiated with a slight variation in initial conditions, can lead to vastly different outcomes. This sensitivity to initial conditions is a hallmark of chaotic systems and is particularly valuable in the context of PRNGs, where unpredictability is a desired feature. Inspired by this, our project aims to leverage the logistic map to develop a PRNG that is not only efficient and dependable but also implementable on a Field-Programmable Gate Array (FPGA), a platform celebrated for its versatility and capability for rapid computations. We code out the different modules we require for this implementation using Verilog HDL on Vivado, and we use a CMOD-A7 FPGA board along with a 7-segment display for testing and visualisation.

This report outlines our journey from exploring the potential of chaotic systems for PRNGs to the detailed implementation of the logistic map on an FPGA, culminating in a system that generates Gaussian-distributed pseudo-random numbers, underscored by the innovative use of a sensor-derived seed. Through this endeavour, we aim to demonstrate the practical utility and effectiveness of chaotic systems, especially the logistic map, in the realm of pseudo-random number generation.

#### 2. Literature Review

In exploring the development of pseudo-random number generators (PRNGs) using chaotic systems, we did a literature review to find insights from several studies on the logistic map, coupled maps, and statistical methods for achieving Gaussian distributions from chaotic outputs. Our focus was on understanding the dynamics of the logistic map, adapting the Central Limit Theorem (CLT) for dependent variables, and employing the Exponentially Weighted Moving Average (EWMA) technique to allow us to correctly make a pseudo-random number generator that followed a Gaussian distribution.



$$x_{n+1} = r x_n (1 - x_n)$$

Equation 1: Logistic Map Function

The logistic map is recognised for its simple yet chaotic behaviour, offering a rich source of unpredictability essential for PRNGs. Its sensitivity to initial conditions, a characteristic feature of chaotic systems, is particularly valuable for generating pseudo-random sequences. It works when r is in the chaotic range, the value of  $x_{n+1}$ , the iterating value of x essentially becomes non-periodic, and does not have a pattern, even though it is still deterministic. This is why we chose the logistic map function for our PRNG.

Coupled maps extend this complexity, suggesting that interactions between multiple chaotic systems can enhance the unpredictability and quality of the generated sequences. This approach aims to leverage the compounded chaos for more robust PRNGs. But in our implementation, coupled maps, or specifically coupled logistic maps would make it way too complex.

Transforming the output from chaotic systems to follow a Gaussian distribution is crucial for many applications. However, the dependency of successive outputs in chaotic systems poses a challenge to traditional methods like the CLT. Research in this area has focused on adapting the CLT to accommodate these dependencies, exploring theoretical frameworks and practical techniques to achieve normally distributed outputs from chaotic sequences.

The EWMA method is highlighted as a tool for mitigating the correlation between successive values in a sequence, facilitating the application of the CLT to sequences generated by chaotic systems. This approach helps in smoothing the data and making it more suitable for generating Gaussian-distributed pseudo-random numbers.

We noted down the references we used for this literature review in the references section of this report.

## 3. PRNG Algorithm

After understanding more about the chaotic systems, CLT and EWMA, we tried to make tests and prototypes before implementing things in Verilog HDL.

The reason for using a chaotic system like the logistic map is mainly due to its deterministic nature and chaotic, unpredictable nature, which is sensitive to an input seed value. These make it ideal for use in a PRNG.

$$x_{n+1} = r x_n (1 - x_n)$$

Equation 1: Logistic Map Function

(In our case, we take r = 4 as a value of r, which makes this chaotic. This works for any other 3.5 < r < 4)

We use the output from the logistic map function and take its value EWMA, as this would allow for the application of CLT, such that the EWMA output is normal. We use EWMA as it is a simple, efficient method that allows the data to be averaged or aggregated so that they may exhibit properties of independence, more so than the raw outputs of the logistic map function. Based on the literature review, we saw that CLT techniques can be applied to datasets that are not exactly independent or that can be applied to moving averages. From this knowledge, we try to use EWMA so that CLT may be applicable in our case.

$$EWMA_t = \alpha r_t + (1 - \alpha) EWMA_{t-1}$$

Equation 2: EWMA Function (In our case, we take  $\alpha$ =40/50 from experimenting with the algorithm in Julia)

In fact, in many real-world applications, especially in finance and economics, EWMA and similar smoothing techniques are used to analyse time series data. For practical purposes, such as risk assessment and forecasting, the resulting smoothed series are often treated as if they were normally distributed. This is why we thought to use the EWMA to make our output conform to a normal or Gaussian distribution.

We did some proof of concepts using code, done in Julia. We found that we can indeed empirically obtain a distribution of PRNG outputs close to a Gaussian Distribution. However, whether it is indeed a Gaussian distribution or not and whether CLT can apply is something that we shall not theoretically prove in this report, as it may take attention away from our original intention. Nonetheless, we provide sources in the references that support the use of chaos theory and EWMA to make a normally distributed PRNG.

### 3.1 Proof Of Concept

For this, we used the Plots package in Julia, and used a simple logistic map function to generate a random number. From here, we used the idea of EWMA to ensure that our final output value for this PRNG would be a value that would conform to a Gaussian Distribution by CLT. We double-check that it follows a Gaussian distribution by plotting a histogram of the PRNG outputs and a Gaussian Curve to overlay it. We also confirm whether the PRNG is "random" by plotting the outputs against time (iterations) in this case, as shown in Figure 2. The values observed over the iterations are not periodic and are hard to predict and determine, making it seem random to those unaware of the algorithm being used.

Code:

```
using Plots
maxiter = 1000 # More iterations for more data
window_size = 50 # Window size for the rolling average
function f(x, r=4); (r * x * (65535 - x) / 65535); end
x0 = 6000; stored_x = zeros(maxiter); stored_x[1] = x0
for n = 2:maxiter; stored_x[n] = f(stored_x[n-1]); end

# Calculate the rolling averages
rolling_averages = zeros(maxiter); rolling_averages[1] = stored_x[1]
for i = 2:maxiter; rolling_averages[i] = (40*rolling_averages[i-1] + 10*stored_x[i])/50; end
```

```
for i = 1:maxiter; rolling_averages[i] = floor( rolling_averages[i]); end

#Plotting the distribution of rolling averages
histogram(rolling_averages, bins=10, label="PRNG Output", xlabel="Value",
ylabel="Frequency", title="Distribution of PRNG", xlims = (0,65535))
using Statistics # For mean()

# Assuming mean_val and std_dev are already calculated
mean_val = mean(rolling_averages); std_dev = std(rolling_averages)

# Manually calculate the PDF values
function normal_pdf(x, μ, σ); return (1.0 / (σ * sqrt(2 * π))) * exp(-0.5 * ((x - μ) / σ)^2); end

# Generate the x values (ensure this covers the range of your data)
x_vals = collect(1:2:65535)

# Calculate the PDF for each x value
normal_dist = [normal_pdf(x, mean_val, std_dev) for x in x_vals]

# Overlay the manually calculated normal distribution
plot!(x_vals, normal_dist, label="Normal Distribution", linewidth=2, colour=:red)

# Run separately this next line to show values against iter to confirm whether random or not
plot(1:maxiter, rolling_averages, label="PRNG Output", ylabel="value", xlabel="iter", title="Evolution of PRNG output over time")
```



Figure 2: Proof Of Concept Plots for PRNG Algorithm

## 3.2 FPGA Implementation of PRNG

The big problem with using this is that, typically, in FPGA, the calculations deal with discrete numbers. Dealing with decimals would suggest the need for many shift registers, making the implementation overly complicated. As such, we rescale the original logistic map function. We cap the possible number of values we want to display at  $2^{16}$ , the total number of values that can be represented by 16 bits.

Thus, we rescaled the logistic map function by 65535 (as seen in the code) and floored each value obtained. Based on the logistic map function, this would always give us a pseudo-random integer between 0 and 65535.

Implementing this in Verilog HDL, we created two functions: chaotic\_Lmap and EWMA\_avg. A seed initiates the first output value, which iteratively passes through the chaotic\_Lmap function. This function gives an input to EWMA, which then returns our desired pseudo-random numbers.

```
23 module EWMA_avg(
                                                        24
                                                             //input wire [15:0] current avg,
                                                        25
                                                             input wire clock,
23 🖨 module chaotic_Lmap(
                                                        26 input wire [15:0] seed,
24
    input clock,
                                                             input wire reset,
25
    //input reset.
                                                        28 input wire [15:0] xt,
26 input [15:0]xt,
                                                            output [15:0] avg
27
    input [7:0] r,
                                                        30
                                                               );
28
    output [15:0]xtnext
                                                        31
                                                            wire [31:0] inter;
       );
                                                        32 !
                                                            assign inter = (avg*40 + 10*xt)/50;
30
                                                        33
                                                            reg [31:0] seedinter;
                                                        34
                                                            reg [15:0] rr;
31
    reg [15:0] xtnext r;
32
    assign xtnext = xtnext_r;
                                                        36 reg [15:0] avg_r;
                                                        37 | assign avg = avg_r;
34 wire [31:0] intermediate;
                                                        38
    assign intermediate = r*xt*(65535-xt);
                                                        39 🖯 always @(posedge xt or negedge reset) begin
36 ⊖ //reg next;
                                                        40 if (!reset) begin
37 \triangle //assign ctnext = next;
                                                                     seedinter = rr*seed*(65535-seed) ;
                                                        41 🖯 //
38 palways @(posedge clock) begin
                                                        42 🗎 //
                                                                      avg = seedinter[15:0];
    // intermediate < = r*xt*(65535-xt);
39
                                                                    avg_r = seed;
        xtnext_r <= (intermediate + 32767) / 65535; 44 \(\hat{\text}\)</pre>
40
                                                        45 🖯 //
                                                                  if xt == sometransformation of seed begin
41 🖨 end
                                                        46 //
                                                                  avg <= xt;
42
                                                        47 A //end
43 A endmodule
                                                        48 🖨
                                                                else begin
                                                                    avg_r <= inter[15:0];
                                                        49
                                                        50 A
                                                        51 🛆 end
```

Figure 3: Verilog HDL Code for PRNG Algorithm



Figure 4: Simulation for PRNG (Output PRN is called "avg" in the simulation)

Essentially, we created the functions, ensuring we were using the correct nets, reg, and wire for each variable and that all the outputs and buffers were assigned correctly. We simulated this to ensure that our logic for the modules was correct before integrating them with the other modules meant for XADC, UART, clock, and 7-segment display.

## 4. Setup of FPGA



Figure 5: Simplified System Architecture for our overall FPGA implementation



Figure 6: Schematic Diagram for CMOD A7 to 4 digit 7 Segment Display & Microphone

We decided on an overall system architecture, taking a top-down approach to designing this PRNG to ensure an organised implementation. Then, we developed each module according to the system architecture.

Each module has a specified purpose and is necessary for the functions of our PRNG.

- **xadc**: A module that reads and effectively outputs the readings obtained from the sensor, which are displayed in 16 bits.
- **chaotic\_Lmap:** This module takes in a seed or initial value from the xadc and iteratively passes the value through the formula for a logistic map. This effectively creates a random variable, as the output is chaotic.
- **EWMA\_avg:** A module used to take in a random value from the chaotic\_Lmap and process it through the EWMA algorithm, outputting a 16-bit number that is effectively random and follows a Gaussian distribution
- **drv\_segment:** A module that takes in the random number to be displayed and displays it as a hex number onto a 7-segment display with 4 digits.
- **Uart\_tx:** A module that transmits the 16-bit number in chunks of 8-bits to a receiver.
- clk1: A 1 Hz clock used for the updating of different values for the seed derived from the xadc module, as well as to update the different variables used in chaotic\_Lmap and EWMA\_avg
- **clk2:** The module drv\_segment drives the 500 Hz clock cycle between the different digits of the 7-segment display.
- clk3, clk4, clk5: Three clocks of different frequencies are essential to the functions of the UART. They cycle between chunks of 8 bits to transmit and set the baud rate for serial communication to the laptop receiver.

For the sensor circuit, we used a mic so that we could take in a seed that is dependent on the volume of the surroundings. The sensor can be swapped easily with other sensors, as the whole PRNG depends only on the internal algorithm.

## 5. Actual Verilog HDL implementation

In this section, we will document the more interesting parts of the code used in the Verilog HDL implementation and the code used to receive and process the data on the receiver side (laptop).

For the PRNG algorithm, we kept it the same as in Figure 2, but the important things to note here are that for the chaotic\_Lmap algorithm, we need its output to feedback to its input, but in Verilog, this gets a bit tricky. A quick workaround for this is having an always loop running in the top module, to always update the input to the chaotic\_Lmap based on its output. Also, note that we use a wire called "pee" to feed the output of the chaotic\_Lmap to the input of the EWMA\_avg.

```
always@(negedge rstn or posedge CLK1Hz)begin
    if(!rstn)begin
        xnext <= seed;
        displayed_number_r <= 16'b0;
        end
    else begin
        xnext <= pee;
        displayed_number_r <= disp;
    end
end

chaotic_Lmap Lmap_u0(CLK1Hz, xnext, r, pee);</pre>
```

Figure 7: Simplified System Architecture for our overall FPGA implementation

For the XADC portion in the top\_module, inside the always@(posedge CLK1Hz) begin loop, we take the data acquired from the microphone, assign it to the seed\_r variable and let it read data every 1 second. This seed\_r variable will then be used to initialise the PRNG. We also changed the .daddr\_in(PIN16\_ADDR) to accommodate the microphone data pin.

Figure 8: Modifications to XADC portion in top\_module

For the top\_module, note that we set the seed's default value to 1 instead of 0. If the seed input to the chaotic\_Lmap is 0, the sequence could potentially collapse, and all the outputs from then on will keep on being 0.

```
always @ (posedge CLK1Hz) begin //ne
// disp <= ADC_data;
if (ADC_data == 16'd0) begin
         seed_r <= 16'd1;
         end
    else begin
        seed_r <= ADC_data;
end
end</pre>
```

Figure 9: Simplified System Architecture for our overall FPGA implementation

For the uart\_tx portion in the top\_module, we removed the disp\_r <= odd ? disp\_r +</pre>
1'b1 : disp\_r; as it generated too many numbers that did not fit the Gaussian Distribution we were aiming for. We also flipped the order in the disp\_r <= odd ? disp\_r + 1'b1 :</pre>
disp r; the numbers would come out as reflected on the 7 segment.

```
always @(negedge rstn2 or posedge clk_data_proc) begin

if (!rstn2) begin

disp_r <= 8'h00;
 uart_tx_data <= 8'h00;
 odd <= 1'b0;

end else begin

disp_r <= disp;
 odd <= ~odd;

//uart_tx_data <= odd ? disp_r[15:8] : disp_r[7:0];
 uart_tx_data <= odd ? disp_r[15:8] : disp_r[7:0];

// disp_r <= odd ? disp_r + 1'b1 : disp_r;

//You can replace cnter with your random number generator;

//e.g. a 16 bits wire [15:0] rng_number;

// cnter <= odd ? rng_number : ; //Allow you to transmit number to PC;
end
end</pre>
```

Figure 10: Modifications to uart tx portion in top module

For the UART receiver function, which will be run on a laptop, we run a simple algorithm that stores each new updated value displayed on the 7-segment display. We use ser.read(2) to process 16 bits of data before running the int.from\_bytes function, which changes the 16-bit number to an integer. Apart from that, we plot a histogram based on the stored values, and we overlay a normal distribution that uses the mean and standard deviation of the data. This allows us to see whether the output random numbers are Gaussian easily.

```
main():
   ser = serial.Serial(port=serial_port,
                      baudrate=9600,
                      parity=parity,
                      stopbits=stop_bits,
                      bytesize=bytesize.
                      timeout=1)
   # Receive 100000 Byte
   prev_rx = 0
   data_cnter = []
   for i in range(15000):
       rx_data = int.from_bytes(ser.read(2), byteorder='little')
       if rx_data != prev_rx:
          data_cnter.append(rx_data)
           prev_rx = rx_data
   print(data_cnter)
   n, bins, patches = plt.hist(data_cnter, bins=10, density=True, alpha=0.6, co
    mu, std = np.mean(data_cnter), np.std(data_cnter)
    xmin, xmax = plt.xlim()
    x = np.linspace(xmin, xmax, num=1000)
    p = norm.pdf(x, mu, std)
    plt.plot(x, p, 'k', linewidth=4)
    plt.show()
```

Figure 11: Python code to receive the numbers from our PRNG via serial communication

#### 6. FPGA Testbenches

We also ran test benches for each module to study and analyse their input and outputs.

Testbench Modules:

#### ewma\_tb:



Figure 12: ewma\_tb testbench simulation file

```
or posedge clock) begin
      (!reset) begin
       xt <= seed; // Initialize xt with seed on reset
   end else begin
       xt <= xtnext; // Update xt with xtnext only when not in reset
end
initial begin
   clock = 0;
   reset = 1'b0; // Assert reset initially
   seed = 16'd3000; // Initialize seed value
   xt = seed; // Initialize xt to a known value
   r = 8'd4; // Set the 'r' parameter for chaotic_Lmap
   #10 reset = 1'b1; // Deassert reset after a short delay
   // The simulation now continues with reset deasserted,
   // allowing normal operation and updates based on xtnext.
   #200; // Run the simulation for a time
   seed = 16'd120;
   reset = 1'b0;
   #10 reset = 1'b1;
   #300:
   $finish; // End the simulation
```

Figure 13: ewma\_tb simulation code

#### clk div tb:



Figure 14: clk\_div\_tb testbench simulation file

```
defparam clock_div_uTEST.FREQ_INPUT = 500_000; // Input frequency to the clock divider
defparam clock_div_uTEST.FREQ_OUTPUT = 100_000; // Output frequency for UART operations
initial begin
    clk = 1'b0;
    rstn = 1'b0;

#5 rstn = 1'b1;

#200 $finish;
end
```

Figure 15: clk\_div\_tb simulation code

#### Lmap\_tb:



Figure 16: Lmap\_tb testbench simulation file

```
always begin
    #5 clock <= ~clock;
    end

always @(posedge clock) begin
    #1 xt <= xtnext;
    end

initial begin
//    xtnext = 16'd0;
    clock = 0;
    xt = 16'd3000;
    r = 8'd4;
    #100;

    xt = 16'd12;
    #100
    $finish;

end</pre>
```

Figure 17: Lmap\_tb simulation code

#### Drv\_segment\_tb:



Figure 18: drv\_segment\_tb testbench simulation file

```
eg clk;
 /reg int
always begin
   #1 clk <= ~clk;
always begin
   #10 bcd_num <= bcd_num + 1;
drv segment drv segment uTEST(
   .displayed number(displayed number),
   .rstn(rstn),
   .clk500hz(clk),
   .bcd_num(bcd_num),
    .an(an),
    .segment(segment)
initial begin
   bcd_num = 16'd1234;
   displayed number = 16'd1234;
   clk = 1'b0;
   rstn = 1'b0;
   #5 rstn = 1'b1;
    #200 $finish;
```

Figure 19: drv\_segment\_tb simulation code

#### 7. Results

In the results section of our project, we demonstrate the effective operation of our pseudo-random number generator (PRNG) implemented on the FPGA. The histograms of the PRNG outputs, which were analysed both on the FPGA and via a connected laptop, consistently displayed a Gaussian distribution. This alignment with a Gaussian model confirms the theoretical predictions and the successful application of the Central Limit Theorem (CLT) to the outputs derived from the logistic map.

The figure below illustrates the distribution over roughly 60 iterations, which went on for 5 minutes of run time in real life. It shows the close fit between the empirical data and the Gaussian curve, thereby validating our approach. This evidence underscores the PRNG's capability to generate highly unpredictable and statistically sound sequences.



Figure 20: Distribution of data received from CMOD XADC to PC via UART

Although we note that achieving a nice fit of the pseudo-random numbers with a Gaussian curve, this fit would be even better if we had the time and resources to run the algorithm for hours, such that it would be a few thousand iterations. This would likely result in a smoother histogram that resembles a Gaussian curve.

We also put in some additional effort that included the design and 3D printing of a case for our whole setup. The mic sensor will be put inside the cone, and a phone will be set up so its speaker is also within the cone. The cone will amplify/concentrate the sound and the mic sensor will be picking up on this amplified sound. Hence, the casing plays 2 roles: 1 to amplify the sound from a phone speaker and another to be used as the seed for our PRNG.



Figure 21: 3D Model of Conical Structure to contain the microphone and CMOD circuitry



Figure 22: Cross Sectional of Conical structure containing the CMOD A7 and microphone



Figure 23-24: Top and Front View of Conical Structure containing CMOD A7-XADC-Mic



Figure 25-29: CMOD A7-XADC-Mic in Conical Structure changing values across 5 minutes

## 8. Reflections and Challenges

Throughout this project, our team encountered challenges that provided significant learning opportunities and insights into the complexities of digital system design. One of the most demanding aspects was integrating the various FPGA modules, including the clock generator, UART interface, XADC, and the 7-segment display driver. Ensuring that all modules operated cohesively required meticulous attention to detail and a deep understanding of each component's role within the system.

Additionally, familiarising ourselves with the syntax and intricacies of Verilog HDL presented a steep learning curve. We faced numerous synthesis and implementation issues from subtle bugs in our initial code. These bugs affected not only the functionality of individual

modules but also their interactions, leading to unexpected behaviours during the integration phase.

Despite these challenges, the project was immensely rewarding. It pushed us to apply theoretical knowledge in a practical setting, enhancing our problem-solving skills and ability to work as a cohesive team under pressure. We also gained hands-on experience with advanced tools and technologies that are crucial in the field of digital systems engineering.

Overall, this project not only tested our technical abilities but also taught us the importance of perseverance and meticulousness in engineering. The skills and insights gained from this experience will undoubtedly influence our approach to complex engineering problems as we progress in our careers.

## 9. Group Image



#### 10. References

- 1. Boeing, G. (2016, October 21). *Chaos theory and the logistic map*. Geoff Boeing. https://geoffboeing.com/2015/03/chaos-theory-logistic-map/#:~:text=This%20range% 20of%20parameters%20represents,nature%20of%20the%20logistic%20equation
- Chen, S., Feng, S., Fu, W., & Zhang, Y. (2021). Logistic map: Stability and entrance to Chaos. *Journal of Physics: Conference Series*, 2014(1), 012009. https://doi.org/10.1088/1742-6596/2014/1/012009
- 3. Davidson, J. (1994). In CLTs for Dependent Processes. essay.
- 4. *Gravity: Analog sound sensor for Arduino*. DFRobot. (n.d.). https://www.dfrobot.com/product-83.html
- 5. Kwak, S. G., & Kim, J. H. (2017). Central limit theorem: The cornerstone of modern statistics. *Korean Journal of Anesthesiology*, 70(2), 144. https://doi.org/10.4097/kjae.2017.70.2.144
- 6. Miao, Y., Ge, L., & Xu, S. (2013). Central limit theorems for moving average processes\*. *Lithuanian Mathematical Journal*, *53*(1), 80–90. https://doi.org/10.1007/s10986-013-9195-7
- 7. Oestreicher, C. (2007). A history of chaos theory. *Dialogues in Clinical Neuroscience*, 9(3), 279–289. https://doi.org/10.31887/dcns.2007.9.3/coestreicher
- 8. Sharma, S. (n.d.). *EWMA what is it, formula, examples, importance*. What Is EWMA. https://www.wallstreetmojo.com/ewma/
- 9. Zanette, D. H., & Samengo, I. (2024). The emergence of the normal distribution in deterministic chaotic maps. *Entropy*, *26*(1), 51. https://doi.org/10.3390/e26010051